/*
商业单位需要容易记忆的电话号码，有一些方法可以让电话号码变得更容易记忆。譬如，可以把电话号码写成单词或短语，如 MON - GLOP 可以代表滑铁卢大学的电话。有时仅仅是把号码的一部分写成单词，如打 310 - GINO 便可向 GINO 比萨饼店定购比萨。另一种让电话号码容易记忆的方法是将数字用一种容易记的方式组合起来，譬如 3 - 10 - 10 - 10 也可以代表 GINO 比萨饼店。

电话号码的标准形式是七位十进制数字，在它的第三位和第四位之间用连字符连接(例如：666 - 1200)。电话的键盘提供了字符与数字之间的映射关系，如下所示：

2----A、B和C
3----D、E和F
4----G、H和I
5----J、K和L
6----M、N和O
7----P、R和S
8----T、U和V
9----W、X和Y

Q 和 Z 没有映射到键盘，而连字符不需要被拨打并且可以根据需要添加和删除。MON - GLOP 的标准形式是 666 - 4567，310 - GINO 的标准形式是310 - 4466，3 - 10 - 10 - 10的标准形式也是 310 - 1010。

如果两个电话号码有相同的标准形式，那么这两个电话号码是相同的。

你所在的公司正在编辑一本当地商业单位的电话簿，作为质量控制流程的一部分，你需要确认在该电话簿中有没有错误的电话号码，以及有没有两个(或两个以上的)商业单位使用相同的电话号码。由于当地只使用了 3 和 6 两个区段，因此电话号码的第一个数字应当永远是 3 或者 6，如果出现了其它数字，就表示这个电话号码错了。此外，如果电话号码中出现了 Q 和 Z，也说明这个电话错了。

输入
一次输入为一个样例。每个号码一行，每行的字符不会超过 20 个。每次输入的数据可能会非常大，譬如超过 1, 000, 000 个电话号码。

你可以假设输入中可能会出现重复的电话号码不超过 1, 500 个，每个号码重复的次数不超过 1000 次。

输出
输出包括两个部分，第一个部分是错误的电话号码，对于这些号码应当按照输入的顺序以原始的形式输出。在输出错误电话号码前输出“Error : ”，随后输出这些号码，如果没有错误的电话号码，则输出“Not found.”。

	第二部分是重复的电话号码，对每一个在电话簿中以任何形式出现一次以上的电话号码，生成一行输出。这一行应以标准形式给出电话号码，其后跟随一个空格，空格后跟随电话号码在电话簿中出现的次数。所有重复的电话号码输出行应以号码的升序排列（小号码在前）。在输出重复电话号码前输出“Duplication”，随后按照上述格式输出号码，如果在输入中没有重复的电话号码，则输出：“Not found.”。

	注意
	你所编写的程序以后可能会在一种特殊的嵌入式设备上运行，为了降低成本，这种设备使用的 CPU 不是很快、可用的 RAM 为 288K（跟 GBA 一样）且它没有磁盘设备因此不能使用文件作为数据的临时存储。

	提示
	请参考《编程珠玑》第一部分，若程序不能在规定的内存中运行，则不得分。
*/


#ifdef _MSC_VER
#define _CRT_SECURE_NO_WARNINGS
#endif
#if defined(_WIN32) && !defined(__cplusplus)
#define inline __inline
#endif

#include <stdio.h>
#include <string.h>

typedef int UINT;

typedef int		BOOL;
#define TRUE	1
#define FALSE	0

#define MAX_NUM 31250 // 1000000 / 32
#define MAX_DUPLICATION_NUM 1500
typedef struct
{
	UINT TeleNum;
	short Times;
}Duplication;

static inline int StrToInt(const char *str)
{
	int len = strlen(str);
	int temp = 0;
	int bit = 1;
	for (int i = len - 1; i >= 0; i--)
	{
		temp += (str[i] - '0') * bit;
		bit *= 10;
	}
	return temp;
}

//存入位图
static inline void SaveInBitmap(UINT bitmap[], UINT num)
{
	bitmap[num >> 5] |= (1 << (num & 0x1f));
}

//检验是否存在
static inline BOOL IsExistent(UINT bitmap[], UINT num)
{
	return bitmap[num >> 5] & (1 << (num & 0x1f));
}

static inline char GetBit(char a)
{
	if (a <= 'O') return (a - 'A') / 3 + '2';
	else if (a >= 'T') return (a - 'T') / 3 + '8';
	else return '7';
}

static inline int cmp(const void *a, const void *b)
{
	return  ((Duplication*)a)->TeleNum - ((Duplication*)b)->TeleNum;
}

UINT GetRealTeleNum(const char *str)
{
	int len = strlen(str);
	char s[20] = {0};
	for (int i = 0, j = 0; i < len; i++)
	{
		char now = str[i];
		if (now == 'Q' || now == 'Z') return 0;
		else if (now >= '0' && now <= '9') s[j++] = now;
		else if (now >= 'A' && now < 'Z') s[j++] = GetBit(now);
	}

	if (s[0] != '3' && s[0] != '6') return 0;
	else return StrToInt(s);
}

int main()
{
	UINT bitmap3[MAX_NUM] = {0};
	UINT bitmap6[MAX_NUM] = {0};
	Duplication duplication[MAX_DUPLICATION_NUM] = {{0,0}};

	char input[20];
	BOOL bError = FALSE;
	int iDupl = 0;

	printf("Error:\n");

	while (scanf("%s", input) != EOF)
	{
		UINT num = GetRealTeleNum(input);
		if (num == 0)
		{
			printf("%s\n", input);
			bError = TRUE;
			continue;
		}

		BOOL bExistent = FALSE;
		if (num > 5999999)
		{
			UINT numSave = num - 5999999;
			if (!IsExistent(bitmap6, numSave)) SaveInBitmap(bitmap6, numSave);
			else bExistent = TRUE;
		}
		else if (num > 2999999)
		{
			UINT numSave = num - 2999999;
			if (!IsExistent(bitmap3, numSave)) SaveInBitmap(bitmap3, numSave);
			else bExistent = TRUE;
		}

		if (!bExistent) continue;

		for (int i = 0; i < MAX_DUPLICATION_NUM; i++)
		{
			if (duplication[i].TeleNum == 0)
			{
				duplication[i].TeleNum = num;
				duplication[i].Times = 2;
				iDupl++;
				break;
			}
			else if (duplication[i].TeleNum == num)
			{
				duplication[i].Times++;
				break;
			}
			else if (duplication[i].TeleNum > num)
			{
				for (int j = iDupl - 1; j >= i; j--)
				{
					duplication[j + 1] = duplication[j];
				}
				duplication[i].TeleNum = num;
				duplication[i].Times = 2;
				iDupl++;
				break;
			}
			else continue;
		}

	}
	if (!bError) printf("Not found.\n");

	printf("\nDuplication:\n");
	if (iDupl == 0)
	{
		printf("Not found.\n");
		return 0;
	}

	for (int i = 0; i < iDupl; i++)
	{
		UINT number = duplication[i].TeleNum;
		printf("%03d-%04d %d\n", number / 10000, number % 10000, duplication[i].Times);
	}

	return 0;
}
